// ==UserScript==
// @name Better NXU
// @namespace https://thisish.com/
// @version 0.2.3
// @description 这是一个增强 NXU 网站使用体验的JavaScript脚本.
// @author H
// @run-at document-idle
// @storageName h.nxu
// @match *://webvpn.nxu.edu.cn*
// @match *://jsfzyjxzlxt.nxu.edu.cn*
// @match *://jwgl.nxu.edu.cn*
// @match *://202.201.128.234*
// @grant unsafewindow
// @grant GM_info
// @grant GM_log
// @grant CAT_userConfig
// @grant GM_addStyle
// @grant GM_getResourceText
// @grant GM_getValue
// @grant GM_setValue
// @grant GM_addElement
// @grant GM_setClipboard
// @grant GM_xmlhttpRequest
// @require https://scriptcat.org/lib/1405/^1.0.6/h.notification.js
// @require https://unpkg.com/tesseract.js@5.1.1/dist/tesseract.min.js
// @require https://unpkg.com/vue@3/dist/vue.global.js
// @resource svg-logo https://cdn.bootcdn.net/ajax/libs/font-awesome/6.2.1/css/all.min.css
// @resource tesseract https://unpkg.com/tesseract.js@5.1.1/dist/tesseract.min.js
// @resource vant-css https://unpkg.com/vant@4/lib/index.css
// @resource vue-js https://unpkg.com/vue@3/dist/vue.global.js
// @resource vant-js https://unpkg.com/vant@4.8.0/lib/vant.min.js
// @connect webvpn.nxu.edu.cn
// ==/UserScript==
// // @require https://cdn.jsdelivr.net/npm/tesseract.js@5/dist/tesseract.min.js
/* ==UserConfig==
WebVPN:
username:
title: 账号
description: 连接校园网的账号
type: text
default: null
password:
title: 密码
description: 连接校园网的密码
type: text
default: null
password: true
autoLogin:
title: 自动登录WebVPN
description: 是否自动登录WebVPN
type: checkbox
default: false
autoClose:
title: 自动关闭错误网站
description: 是否自动关闭显示错误的网站
type: checkbox
default: false
courseGrab:
title: 抢课备用列表
description: 是否添加抢课备用列表
type: checkbox
default: false
customCard:
title: 在主页需要添加的卡片
description: 这里可以选择在主页增加的自定义卡片
type: mult-select
default: ['学工系统','中国知网']
values: ['教务管理','中国知网', '万方数据','学工系统']
qualityJson:
title: 评教系统自定义配置
description: 这里可以配置评教系统的自定义设置,请严格按照以下要求:1. 每一条规则均用[]表示,每条规则之间用英文逗号,隔开。2. 内有三个参数,每个参数之间用英文逗号,隔开。3. 参数1为一个数字,表示第几列;参数2为一个字符串,需用英文单引号'引用,表示这一列匹配的内容是什么;参数3为一个数字,1表示完全同意,2表示同意,以此类推。 示例:[[0, '“四史”教育', 2], [1, 'XX学院', 1]]
type: textarea
default: []
Jwgl:
username:
title: 账号
description: 登录教务系统的账号(一般同校园网)
type: text
default: null
password:
title: 密码
description: 登录教务系统的密码(一般同校园网)
type: text
default: null
password: true
autoLogin:
title: 自动登录教务系统
description: 是否自动登录教务系统(抢课时可解放双手)
type: checkbox
default: false
courseBeautify:
title: 课表美化
description: 是否自动美化课表
type: checkbox
default: false
==/UserConfig== */
(async function() {
'use strict';
// ==Basic==
function Basic() {
// 添加Notification组件
// 添加组件
addToast();
//createToast("success", "测试消息", 0);
// 添加css样式
GM_addStyle(ToastCss);
GM_addStyle(GM_getResourceText("svg-logo").replace(/\.\.\/webfonts/g, "https://cdn.bootcdn.net/ajax/libs/font-awesome/6.2.1/webfonts"));
//绑定Toast事件
unsafeWindow.createToast = createToast;
unsafeWindow.removeToast = removeToast;
// 绑定事件
unsafeWindow.CAT_userConfig = CAT_userConfig;
// UI
unsafeWindow.Vue = Vue;
GM_addStyle(GM_getResourceText("vant-css"));
unsafeWindow.eval(GM_getResourceText("vant-js"));
unsafeWindow.vant = vant;
}
// /==Basic==
// ==Constant==
const Info = GM_info;
const Url = window.location.href;
const Host = window.location.host;
const Origin = window.location.origin;
const Path = window.location.pathname;
const LoadMessage = { "loading tesseract core": "核心加载", "initializing tesseract": "初始化", "loading language traineddata": "加载语言训练数据", "initializing api": "初始化接口", "recognizing text": "识别验证码" };
const ConfigVersion = 1;
// /==Constant==
// ==Function==
function GetVerificationCode(web) {
var url = "";
switch (web) {
case "WebVPN":
url = "https://webvpn.nxu.edu.cn/https/77726476706e69737468656265737421f9f352d229287d1e7b0c9ce29b5b/authserver/captcha.html?vpn-1&ts=225";
break;
case "Jwgl":
url = "captcha/image.action";
break;
default:
return;
}
MyConsole(url)
return new Promise(async function (resolve, reject) {
Tesseract.recognize(
url,
'eng',
{ logger: m => LoadMessage[m.status] ? (MyConsole(LoadMessage[m.status])) : (null) }
).then(({ data: { text } }) => {
MyConsole(text.replace(/\s+/g, ''));
resolve(text.replace(/\s+/g, ''));
})
});
}
function MyConsole(msg) {
if (typeof (msg) != 'object') {
// if (/\n/.test(msg)) {
// console.log("======== Better NXU ========\n" + msg + "\n======== Better NXU ========");
// } else {
console.log(
'%c %s %c %s',
'border-radius: 5px;padding: 3px 4px;color: white;background-color: #3a8bff;margin-bottom: 0.5em',
'Better NXU',
'margin-left: 0.6em;font-size:1.2em',
'\n' + msg,
);
// }
} else {
console.log(
'%c %s %c %s',
'border-radius: 5px;padding: 3px 4px;color: white;background-color: #3a8bff;margin-bottom: 0.5em',
'Better NXU',
'margin-left: 0.6em;',
'\n下面是一个object对象',
);
console.log(msg);
}
}
function CheckUsernameAndSecret(web) {
const username = GM_getValue(web + ".username", false);
const password = GM_getValue(web + ".password", false);
if (!username || !password) {
MyConsole("账号密码未配置\n请前往配置相关信息");
createToast("error", `
账号密码未配置
请前往配置相关信息
> 前往配置 <
`);
return false;
}
return true;
}
function GetQuery(msg) {
// 获取当前页面的 URL
let urlString = window.location.href;
// 创建 URL 对象
let url = new URL(urlString);
// 获取查询参数
let searchParams = new URLSearchParams(url.search);
let result = searchParams.get(msg);
return result;
}
// 随机数
function random(min, max) {
return parseInt(Math.random() * (max - min + 1) + min, 10);
}
// 等待执行
function WaitTime(min, max = 0, log = true, msg = "无") {
var waitmsg, waittime, line;
if (max == 0) {
waittime = min;
waitmsg = `====================\n等待了:${(waittime / 1000)} 秒\n备注:${msg}\n====================`;
} else {
waittime = random(min, max);
waitmsg = `====================\n随机等待了:${(waittime / 1000)} 秒\n备注:${msg}\n====================`;
}
return new Promise(function (resolve, reject) {
setTimeout(function () {
if (log) {
MyConsole(waitmsg.replace(/ /g, ""));
}
resolve();
}, waittime);
});
}
// /==Function==
MyConsole(`开始运行`);
MyConsole(Info);
while (document.readyState != "complete") {
await WaitTime(100);
}
MyConsole(`判断页面...`);
switch (Host) {
case 'webvpn.nxu.edu.cn':
MyConsole("欢迎使用 webvpn");
if (Url.indexOf("service=https%3A%2F%2Fwebvpn.nxu.edu.cn%2Flogin%3Fcas_login%3Dtrue") != -1) {
MyConsole("这里是 - 登录页");
unsafeWindow.eval(GM_getResourceText("tesseract"));
Tesseract = unsafeWindow.Tesseract;
Basic();
webvpnLogin();
} else if (Url == 'https://webvpn.nxu.edu.cn/') {
MyConsole("这里是 - 主页");
Basic();
webvpnMain();
} else if (Url.indexOf('xsfw/sys/xggzptapp/*default/index.do')) {
MyConsole("这里是 - 学工系统");
} else if (Url.indexOf('/77726476706e69737468656265737421fae04690693e7045300d8db9d6562d/') != -1 || Url.indexOf('/77726476706e69737468656265737421a2a713d27560391e2f5ad1e2ca0677/') != -1) {
MyConsole("这里是 - 教务系统");
if (Url.indexOf('index.action') != -1 || Url.indexOf('login.action') != -1) {
MyConsole("正在 - 登录页");
unsafeWindow.eval(GM_getResourceText("tesseract"));
Tesseract = unsafeWindow.Tesseract;
Basic();
jwglLogin();
} else if (Url.indexOf('cas.action') != -1) {
MyConsole("正在 - 主页");
Basic();
jwglMain();
} else if (Url.indexOf('courseTableForStd.action') != -1 && GetQuery('method') == 'stdHome') {
MyConsole("正在 - 课表");
jwglCourseIframe();
} else if (Url.indexOf('courseTableForStd.action') != -1 && GetQuery('method') == 'courseTable') {
MyConsole("正在 - 课表");
jwglCourseBeautify();
}
} else if (Url.indexOf('/77726476706e69737468656265737421fbf952d2243e635930068cb8') != -1 || Url.indexOf('/77726476706e69737468656265737421e7e056d2243e635930068cb8') != -1) {
MyConsole("这里是 - 中国知网");
if (Url.indexOf('/xmlRead/trialRead') != -1) {
MyConsole("正在 - html阅读");
webvpnCnkiHtml();
}
}
break;
case 'jsfzyjxzlxt.nxu.edu.cn':
MyConsole("欢迎使用评教系统");
if (Path == "/quality/student/evaluate/item_tasks") {
MyConsole("这里是 - 选择页");
qualityChoose();
} else if (Path == "/quality/student/evaluate/item_tasks_text") {
MyConsole("这里是 - 填写页");
qualityText();
}
break;
case 'jwgl.nxu.edu.cn':
MyConsole("欢迎使用教务系统");
if (Path == '/index.action' || Path == '/login.action') {
MyConsole("这里是 - 登录页");
Basic();
jwglLogin();
} else if (Url.indexOf('cas.action') != -1) {
MyConsole("正在 - 主页");
Basic();
jwglMain();
} else if (Url.indexOf('courseTableForStd.action') != -1 && GetQuery('method') == 'stdHome') {
MyConsole("正在 - 课表");
jwglCourseIframe();
} else if (Url.indexOf('courseTableForStd.action') != -1 && GetQuery('method') == 'courseTable') {
MyConsole("正在 - 课表");
jwglCourseBeautify();
}
// case 'tuanwei.nxu.edu.cn':
// MyConsole("欢迎使用团委");
// if (Path == '/system/_content/download.jsp') {
// MyConsole("这里是 - 附件下载页");
// tuanweiDownload();
// } else {
// tuanweiSearch();
// }
default:
if (Host.indexOf('202.201.128.234') != -1) {
MyConsole("欢迎使用教务系统");
if (Path == '/index.action' || Path == '/login.action') {
MyConsole("这里是 - 登录页");
Basic();
jwglLogin();
} else if (Url.indexOf('cas.action') != -1) {
MyConsole("正在 - 主页");
Basic();
jwglMain();
} else if (Url.indexOf('courseTableForStd.action') != -1 && GetQuery('method') == 'stdHome') {
MyConsole("正在 - 课表");
jwglCourseIframe();
} else if (Url.indexOf('courseTableForStd.action') != -1 && GetQuery('method') == 'courseTable') {
MyConsole("正在 - 课表");
jwglCourseBeautify();
}
}
return;
}
return;
async function webvpnLogin() {
if (!GM_getValue("WebVPN.autoLogin", false)) {
return;
}
createToast("info", `自动登录...`);
if (!CheckUsernameAndSecret("WebVPN")) {
return;
}
if (document.querySelector('span#msg.auth_error') && document.querySelector('span#msg.auth_error').innerHTML && document.querySelector('span#msg.auth_error').innerHTML == '您提供的用户名或者密码有误') {
createToast("error", `
账号密码配置错误
请前往配置相关信息
> 前往配置 <
`);
return;
}
if (document.querySelector("p#cpatchaDiv") && document.querySelector("p#cpatchaDiv").innerHTML && document.querySelector("p#cpatchaDiv").innerHTML.replace(/\s/g, '') != '') {
// var verification = await GetVerificationCode("WebVPN");
// document.querySelector('input#captchaResponse').value = verification;
createToast("warning", `请手动输入验证码登录`, 0);
return;
}
document.querySelector('input#username').value = GM_getValue("WebVPN.username");
document.querySelector('input#password').value = GM_getValue("WebVPN.password");
document.querySelector('button[type=submit]').click();
}
async function jwglLogin() {
if (!GM_getValue("Jwgl.autoLogin", false)) {
return;
}
createToast("info", `自动登录...`);
if (!CheckUsernameAndSecret("Jwgl")) {
return;
}
if (document.querySelector('div#errors.message') && document.querySelector('div#errors.message').innerHTML) {
const errorText = document.querySelector('div#errors.message').innerHTML;
if (errorText == "密码错误" || errorText == "账户不存在") {
createToast("error", `
账号密码配置错误
请前往配置相关信息
> 前往配置 <
`);
return;
}
}
var verification = await GetVerificationCode("Jwgl");
document.getElementsByName("loginForm.name")[0].value = GM_getValue("Jwgl.username");
document.getElementsByName("loginForm.password")[0].value = GM_getValue("Jwgl.password");
document.getElementsByName("loginForm.captcha")[0].value = verification;
document.querySelector("input#loginSubmit").click();
}
async function webvpnMain() {
const firstSet = GM_getValue('firstSet', 0);
const configVersion = GM_getValue('configVersion', -1);
// createToast("success", "测试消息", 0);
//更新配置弹窗
MyConsole("检查配置信息");
GM_addElement(document.querySelector('body'), 'div', { id: 'update' });
var update_template = '', update_show = false;
if (!firstSet) {
update_template = `
这好像是你第一次使用本插件
我们需要一些配置信息
你可以选择前往配置
或点击取消不进行配置
后续自行前往设置页面进行配置
`;
update_show = true;
} else if (configVersion != ConfigVersion) {
update_template = `
V ${Info.script.version}
我们更新了一些配置信息
你可以选择前往配置
或点击取消不进行配置
后续自行前往设置页面进行配置
`;
update_show = true;
}
const update = Vue.createApp({
template: update_template,
setup() {
const show = Vue.ref(false);
show.value = update_show;
const goConfig = () => {
CAT_userConfig();
}
const closeFunc = () => {
GM_setValue('firstSet', true);
GM_setValue('configVersion', ConfigVersion);
}
return { show, goConfig, closeFunc };
}
});
update.use(vant);
update.mount("#update");
// //更新弹窗
// const version = GM_getValue('version');
// if (version != Version) {
// GM_setValue('version', Version);
// createToast("info", `
// Better NXU V ${Version} 已更新!
// ${VersionContent}
// > 前往配置 <
// `);
// }
// 抢课备用列表
function divCard(href, icon, title, content) {
var div = document.createElement('div');
div.innerHTML = `
${icon}
`;
div.className = 'block-group__item__wrap';
return div;
}
function titleCard(title, id) {
var div = document.createElement('div');
div.innerHTML = `
${title}
`;
div.className = 'block-group';
div.dataset.id = id;
return div;
}
const mainDiv = document.querySelector('.portal-content__block .el-scrollbar__view');
if (GM_getValue('WebVPN.courseGrab', false)) {
mainDiv.prepend(titleCard("抢课备用网址", "classes"));
const classesDiv = document.querySelector('div[data-id=classes] div.block-group__content');
for (let i = 0; i <= 3; i++) {
classesDiv.appendChild(divCard(
`http://202.201.128.234:${8080 + i}`,
'抢
',
`备用${i + 1}`, "仅校园网可用")
);
}
for (let i = 0; i <= 3; i++) {
classesDiv.appendChild(divCard(
`https://webvpn.nxu.edu.cn/http-${8080 + i}/77726476706e69737468656265737421a2a713d27560391e2f5ad1e2ca0677/index.action`,
'抢
',
`备用${i + 5}`, "校外可用")
);
}
}
//自定义卡片
const webVPNCustomCard = GM_getValue("WebVPN.customCard");
if (webVPNCustomCard.length != 0) {
mainDiv.prepend(titleCard("自定义", "custom"));
const customDiv = document.querySelector('div[data-id=custom] div.block-group__content');
if (webVPNCustomCard.indexOf('教务管理') != -1) {
customDiv.appendChild(divCard(
`https://webvpn.nxu.edu.cn/https-443/77726476706e69737468656265737421fae04690693e7045300d8db9d6562d/cas.action`,
'教
',
`教务平台`, "教务管理平台")
);
}
if (webVPNCustomCard.indexOf('学工系统') != -1) {
customDiv.appendChild(divCard(
`https://webvpn.nxu.edu.cn/https/77726476706e69737468656265737421e8e4478b693e7045300d8db9d6562d/`,
'学
',
`学工系统`, "学工平台")
);
}
if (webVPNCustomCard.indexOf('中国知网') != -1) {
customDiv.appendChild(divCard(
`https://webvpn.nxu.edu.cn/https/77726476706e69737468656265737421e7e056d2243e635930068cb8/`,
'知
',
`中国知网`, "中国期刊全文数据库")
);
}
if (webVPNCustomCard.indexOf('万方数据') != -1) {
customDiv.appendChild(divCard(
`https://webvpn.nxu.edu.cn/https/77726476706e69737468656265737421e7e056d2303166567f068ea89941227bfcd3ca21bd0c/`,
'万
',
`万方数据`, "万方数据知识服务平台")
);
}
}
}
async function jwglCourseIframe() {
// 获取 iframe 元素
var iframe = document.querySelector("#contentListFrame");
while (iframe.contentDocument && iframe.contentDocument.readyState != "complete") {
await WaitTime(100);
}
// 等待 iframe 中的内容加载完成后获取内容高度并设置 iframe 高度
var iframeDocument = iframe.contentDocument || iframe.contentWindow.document;
var height = iframeDocument.body.scrollHeight + 'px';
iframe.style.height = height;
}
async function jwglCourseBeautify() {
if (!GM_getValue('Jwgl.courseBeautify')) {
return;
}
function jwglClass(content_array, mode = -1) {
if (mode == 0) {
return `
${content_array[3]}
${content_array[1]}
${content_array[2]}
${content_array[0]}
`;
} else if (mode > 0) {
return `
${content_array[2]}
${content_array[0]}
`;
} else {
return `
${content_array[3]}
${content_array[0]}
${content_array[2]}
${content_array[1]}
`;
}
};
GM_addStyle(`
`);
$("body").html($("body").html().replace(".noneprint{\n\tdisplay:none\n} \n\n", ""))
$("table.listTable#contentListFrame").addClass('optimized');
$("table").css("margin-bottom", "3em");
$("tr").attr("height", "auto");
$("tr").css("min-height", "45px");
$("td").css("padding-left", "0");
$("td > div").each(function () {
var div = $(this);
//div.html(div.attr('title'));
div.css("height", "auto");
div.css("padding", "1em 0.5em 0");
var content = div.attr('title');
// content = content.split(/(? 1) {
if (i == 0 || content_array[i][1] == content_array[i - 1][1]) {
div.append(jwglClass(content_array[i], i));
} else {
div.append('')
div.append(jwglClass(content_array[i], 0));
}
} else {
div.append(jwglClass(content_array[i]));
}
}
});
}
async function webvpnCnkiHtml() {
function getCurrentSelect() {
let selectionObj = null, rangeObj = null;
let selectedText = "", selectedHtml = "";
// 处理兼容性
if (window.getSelection) {
// 现代浏览器
// 获取text
selectionObj = window.getSelection();
selectedText = selectionObj.toString();
// 获取html
rangeObj = selectionObj.getRangeAt(0);
var docFragment = rangeObj.cloneContents();
var tempDiv = document.createElement("div");
tempDiv.appendChild(docFragment);
selectedHtml = tempDiv.innerHTML;
} else if (document.selection) {
// 非主流浏览器IE
selectionObj = document.selection;
rangeObj = selectionObj.createRange();
selectedText = rangeObj.text;
selectedHtml = rangeObj.htmlText;
}
return {
text: selectedText,
html: selectedHtml
}
};
// 使标题可选中
document.querySelector('h1.Chapter').setAttribute('style', 'user-select:auto;');
MyConsole("复制已开启");
// 监听内容区域鼠标抬起事件
document.addEventListener('mouseup', function () {
var copy = getCurrentSelect();
if (copy.text == "") {
return;
}
// myConsole('onmouseup');
MyConsole(getCurrentSelect());
GM_setClipboard(getCurrentSelect().text);
});
}
})();